linux下system函数调用shell命令后,怎样让主进程等子进程返回后,接着执行 您所在的位置:网站首页 shell 调用函数 linux下system函数调用shell命令后,怎样让主进程等子进程返回后,接着执行

linux下system函数调用shell命令后,怎样让主进程等子进程返回后,接着执行

2023-04-24 14:10| 来源: 网络整理| 查看: 265

这是进程间同步的问题。解决方法是:fork一个子进程执行system调用,父进程调用 wait 或 waitpid 等待子进程的终止信息。

父进程调用 wait 或 waitpid 时可能会:

• 阻塞(如果它的所有子进程都还在运行)。

• 带子进程的终止信息立即返回(如果一个子进程已终止,正等待父进程读取其终止信息)。

• 出错立即返回(如果它没有任何子进程)。

wait 和 waitpid 这两个函数的区别是:

• 如果父进程的所有子进程都还在运行,调用wait将使父进程阻塞,而调用waitpid时如果在options参数中指定WNOHANG可以使父进程不阻塞而立即返回0。

• wait等待第一个终止的子进程,而waitpid可以通过pid参数指定等待哪一个子进程。

Linux中启动另一个可执行文件或程序用system函数最理想了,这个函数将在你编写的那个程序的内部启动另一个程序,从而创建一个新进程,并等待这个进程执行完毕退出。如果正常执行,system函数将返回被执行程序的退出码;如果无法运行这个程序,将返回错误代码127;如果是其他错误,返回-1。这个函数的原型是:

#include stdlib.h

int system(const char *string)

参数string是将要执行的程序的命令字符串。

还有一种执行外部程序的方法是exec系列函数,但这个系列的函数都是将当前进程的替换成新进程,也就是说原来的进程不存在了。

linux c system函数介绍:

system(执行shell 命令)

相关函数

fork,execve,waitpid,popen

表头文件

#i nclude

定义函数

int system(const char * string)

函数说明

system()会调用fork()产生子进程,由子进程来调用/bin/sh-c string来执行参数string字符串所代表的命令,此命令执行完后随即返回原调用的进程。在调用system()期间SIGCHLD 信号会被暂时搁置,SIGINT和SIGQUIT 信号则会被忽略。

返回值

=-1:出现错误

=0:调用成功但是没有出现子进程

0:成功退出的子进程的id

如果system()在调用/bin/sh时失败则返回127,其他失败原因返回-1。若参数string为空指针(NULL),则返回非零值。如果system()调用成功则最后会返回执行shell命令后的返回值,但是此返回值也有可能为 system()调用/bin/sh失败所返回的127,因此最好能再检查errno 来确认执行成功。

附加说明

在编写具有SUID/SGID权限的程序时请勿使用system(),system()会继承环境变量,通过环境变量可能会造成系统安全的问题。

范例

#i nclude

main()

{

system("ls -al /etc/passwd /etc/shadow")

}

执行结果:

-rw-r--r-- 1 root root 705 Sep 3 13 :52 /etc/passwd

-r--------- 1 root root 572 Sep 2 15 :34 /etc/shado

例2:

char tmp[]

sprintf(tmp,"/bin/mount -t vfat %s /mnt/usb",dev)

system(tmp)

其中dev是/dev/sda1。

system函数的源码

#include syspes.h

#include sys/wait.h

#include errno.h

#include unistd.h

int system(const char * cmdstring)

{

pid_t pid

int status

if(cmdstring == NULL){

return (1)

}

if((pid = fork())0){

status = -1

}

else if(pid = 0){

execl("/bin/sh", "sh", "-c", cmdstring, (char *)0)

-exit(127)//子进程正常执行则不会执行此语句

}

else

{

while(waitpid(pid, status, 0) 0){

if(errno != EINTER)

{

status = -1

break

}

}

}

return status

}

那么如何获得system的返回值呢??

char buf[10]

char * ps="ps -ef|grep -c root"

FILE *ptr

int i

if((ptr = popen(ps, "r")) != NULL)

{

fgets(buf, 10 , ptr)

i = atoi(buf)

pclose(ptr)

}

可以man下waitpid查看下如何检查status的值

int ret = system("ls -al /etc/passwd /etc/shadow")

if(WIFSIGNALED(ret))

具体的这些宏查看man waitpid

欢迎分享,转载请注明来源:内存溢出

原文地址:https://outofmemory.cn/yw/8395107.html



【本文地址】

公司简介

联系我们

今日新闻

    推荐新闻

    专题文章
      CopyRight 2018-2019 实验室设备网 版权所有